[id].vue 77 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300
  1. // 商品详情页面
  2. <template>
  3. <div>
  4. <StoreHeaderCat ref="headercat" @updateFllow="updateFllow" />
  5. <div class="sld_goods_detail">
  6. <!-- 内容区顶部固定 start -->
  7. <div class="contain_con" v-if="containCon">
  8. <div class="contain_content flex_row_center_center">
  9. <div class="contain_con_left flex_row_start_center">
  10. <span
  11. class="store_type"
  12. v-if="goodsDetail.data.storeInf.isOwnStore == 1"
  13. >{{ L["自营"] }}</span
  14. >
  15. <span class="store_title">{{
  16. goodsDetail.data.storeInf.storeName
  17. }}</span>
  18. </div>
  19. <div class="contain_con_right">
  20. <div class="goods_description_title flex_row_between_center">
  21. <div class="description_title_left flex_row_start_center">
  22. <span
  23. class="cursor_pointer"
  24. :class="{ description_active: goodsDesctionType == 'detail' }"
  25. @click="goodsDescType('detail')"
  26. >{{ L["商品详情"] }}</span
  27. >
  28. <span
  29. class="cursor_pointer"
  30. :class="{
  31. description_active: goodsDesctionType == 'evaluate',
  32. }"
  33. @click="goodsDescType('evaluate')"
  34. >{{ L["商品说明"] }}</span
  35. >
  36. <span
  37. class="cursor_pointer"
  38. :class="{
  39. description_active: goodsDesctionType == 'service',
  40. }"
  41. @click="goodsDescType('service')"
  42. >{{ L["商品服务"] }}</span
  43. >
  44. <span
  45. class="cursor_pointer"
  46. :class="{
  47. description_active: goodsDesctionType == 'salestore',
  48. }"
  49. @click="goodsDescType('salestore')"
  50. >{{ L["店铺热销"] }}</span
  51. >
  52. </div>
  53. </div>
  54. </div>
  55. </div>
  56. </div>
  57. <!-- 内容区顶部固定 end -->
  58. <div class="goods_detail_content self_background" v-if="goodsDetail.data">
  59. <!-- 商品所属分类 ,联系客服,关注店铺 start-->
  60. <div class="goods_about_con">
  61. <div class="goods_about flex_row_between_center">
  62. <div class="goods_classify">
  63. <span
  64. v-for="(item, index) in goodsDetail.data.categoryPath"
  65. :key="index"
  66. >
  67. <nuxt-link
  68. :to="`/goods/list/${calcProductName(item)}_v-${ goodsDetail.data.categoryIdPath[index] }_gid-${index + 1}${index > 0? '_pid-' + goodsDetail.data.categoryIdPath[index - 1]: '_pid-0'
  69. }`"
  70. target="_blank"
  71. >
  72. {{ item }}</nuxt-link
  73. >
  74. <i> > </i>
  75. </span>
  76. <span style="font-weight: bold;font-size: 12px;color: #036EB8;">{{
  77. goodsDetail.data.goodsName
  78. }}</span>
  79. </div>
  80. <div class="goods_about_right flex_row_between_center">
  81. <router-link
  82. target="_blank"
  83. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  84. class="goods_about_store flex_row_between_center"
  85. v-if="goodsDetail.data.storeInf"
  86. >
  87. <span>{{ goodsDetail.data.storeInf.storeName }}</span>
  88. <span
  89. v-if="
  90. goodsDetail.data.storeInf &&
  91. goodsDetail.data.storeInf.isOwnStore == '1'
  92. "
  93. >{{ L["自营"] }}</span
  94. >
  95. </router-link>
  96. <div
  97. class="contact_service focus_store"
  98. @click="focusStore"
  99. v-if="goodsDetail.data.storeInf"
  100. >
  101. <span>
  102. {{
  103. goodsDetail.data.storeInf.isFollowStore
  104. ? L["取消关注"]
  105. : L["关注店铺"]
  106. }}
  107. </span>
  108. <img
  109. src="/goods/collection.png"
  110. alt=""
  111. v-show="goodsDetail.data.storeInf.isFollowStore == true"
  112. />
  113. <img
  114. src="/goods/no_collection.png"
  115. alt=""
  116. v-show="goodsDetail.data.storeInf.isFollowStore == false"
  117. />
  118. </div>
  119. </div>
  120. </div>
  121. </div>
  122. <!-- 商品所属分类 ,联系客服,关注店铺 end-->
  123. <!-- 商品主要信息 start -->
  124. <div class="main_con">
  125. <!-- 商品相关 start -->
  126. <div class="goods_des">
  127. <!-- 商品图片列表 start -->
  128. <div class="goods_des_left">
  129. <!-- 商品放大镜效果 start-->
  130. <div class="goods_main_picture">
  131. <div class="mask" v-if="[5, 6].includes(goodsDetail.data.state)">
  132. <div class="circle">{{ L['已下架'] }}</div>
  133. </div>
  134. <div
  135. class="preview-box"
  136. @mousemove="move($event)"
  137. @mouseleave="out($event)"
  138. @mouseenter="enter($event)"
  139. ref="previewBox"
  140. >
  141. <div
  142. class="imageBorder"
  143. :class="{
  144. default_image: true,
  145. skeleton_default_image: firstLoading,
  146. }"
  147. :style="{ backgroundImage: 'url(' + defaultImage + ')' }"
  148. >
  149. <video
  150. v-if="
  151. currentDefaultImage == -1 && goodsDetail.data.goodsVideo
  152. "
  153. controls
  154. playsinline="playsinline"
  155. class="imageBorder default_image"
  156. :poster="defaultImage"
  157. autoplay
  158. ref="video"
  159. >
  160. <source
  161. :src="goodsDetail.data.goodsVideo"
  162. type="video/mp4"
  163. />
  164. </video>
  165. </div>
  166. <div
  167. class="v_btn"
  168. v-if="
  169. currentDefaultImage != -1 && goodsDetail.data.goodsVideo
  170. "
  171. >
  172. <img src="/goods/playV.png" alt="" @click="playV" />
  173. </div>
  174. <!-- 遮罩 start-->
  175. <div class="mask" ref="maskBox" v-show="maskShow"></div>
  176. <!-- 遮罩 end -->
  177. <!-- 底部放大镜icon图标 start -->
  178. <div
  179. class="magnifier_icon flex_row_center_center"
  180. v-show="!maskShow"
  181. >
  182. <i class="iconfont icon-sousuo"></i>
  183. </div>
  184. <!-- 底部放大镜icon图标 end -->
  185. </div>
  186. <!-- 右侧的放大后的图片 start -->
  187. <div
  188. class="goods_picture_big"
  189. style="border: 1px solid #eee"
  190. ref="zoomBox"
  191. v-show="maskShow"
  192. >
  193. <div
  194. class="default_image_big"
  195. :style="{ backgroundImage: 'url(' + defaultImage + ')' }"
  196. ref="pictureBig"
  197. ></div>
  198. </div>
  199. <!-- 右侧的放大后的图片 end -->
  200. </div>
  201. <!-- 商品放大镜效果 end -->
  202. <!-- 商品图片列表 start -->
  203. <div
  204. :class="{
  205. goods_picture_con: true,
  206. flex_row_between_center: true,
  207. skeleton_goods_picture_con: firstLoading,
  208. }"
  209. v-if="
  210. goodsDetail.data.defaultProduct &&
  211. goodsDetail.data.defaultProduct.goodsPics &&
  212. goodsDetail.data.defaultProduct.goodsPics.length > 0
  213. "
  214. >
  215. <i
  216. class="iconfont icon-ziyuan2 left_arrow"
  217. :class="{ no_left_arrow: currentDefaultImage == 0 }"
  218. @click="switchDefaultImage('left')"
  219. ></i>
  220. <div class="show_box">
  221. <ul
  222. class="goods_picture_list flex_row_start_center"
  223. ref="goodsPictureList"
  224. >
  225. <li
  226. v-for="(goodsImgItem, goodsImgIndex) in goodsDetail.data
  227. .defaultProduct.goodsPics"
  228. :key="goodsImgIndex"
  229. class="goods_img"
  230. :class="{
  231. goods_img_active: currentDefaultImage == goodsImgIndex,
  232. }"
  233. @click="selectDefaultImage(goodsImgItem, goodsImgIndex)"
  234. @mouseover="
  235. selectDefaultImage(goodsImgItem, goodsImgIndex)
  236. "
  237. >
  238. <div
  239. class="goods_image"
  240. :style="{
  241. backgroundImage: 'url(' + goodsImgItem + ')',
  242. }"
  243. ></div>
  244. </li>
  245. </ul>
  246. </div>
  247. <i
  248. class="iconfont icon-ziyuan11 right_arrow"
  249. :class="{
  250. no_left_arrow:
  251. currentDefaultImage ==
  252. goodsDetail.data.defaultProduct.goodsPics.length - 1,
  253. }"
  254. @click="switchDefaultImage('right')"
  255. ></i>
  256. </div>
  257. <!-- 商品图片列表 end -->
  258. <!-- 商品分享和收藏 start -->
  259. <!-- <div
  260. class="collection_share_btn flex_row_start_start"
  261. v-if="goodsDetail.data.state == 3"
  262. >
  263. <div
  264. class="collection_btn flex_row_start_center cursor_pointer"
  265. @click="collectGoods"
  266. >
  267. <img
  268. src="/goods/collection.png"
  269. alt=""
  270. v-if="goodsDetail.data.followGoods"
  271. />
  272. <img src="/goods/collection1.png" alt="" v-else />
  273. <span>{{
  274. goodsDetail.data.followGoods ? L["已收藏"] : L["收藏"]
  275. }}</span>
  276. </div>
  277. <div class="share_btn">
  278. <AddThis publicId="ra-5ab34ca22008ed41" />
  279. </div>
  280. </div> -->
  281. <!-- 商品分享和收藏 end -->
  282. </div>
  283. <!-- 商品图片列表 end -->
  284. <!-- 商品详细信息 start -->
  285. <div class="m_item_inner">
  286. <div class="item_info">
  287. <div
  288. :class="{
  289. detaile_name: true,
  290. skeleton_detaile_name: firstLoading,
  291. }"
  292. >
  293. {{ goodsDetail.data.goodsName }}
  294. </div>
  295. <div
  296. :class="{ p_ad: true, skeleton_p_ad: firstLoading }"
  297. v-if="goodsDetail.data.goodsBrief"
  298. >
  299. {{ goodsDetail.data.goodsBrief }}
  300. </div>
  301. <!-- 商品未下架即正常商品 start -->
  302. <div v-if="goodsDetail.data.state == 3 || firstLoading">
  303. <div class="summary">
  304. <div
  305. class="sld_summary_item summary_goods clearfix"
  306. v-if="goodsDetail.data.defaultProduct"
  307. >
  308. <div class="sld_summary_goods_left">
  309. <div class="goods_price flex_row_start_center">
  310. <!-- 在售价 -->
  311. <div>
  312. <span class="price_title">{{ L["价格"] }}</span>
  313. <strong
  314. :class="{
  315. p_price: true,
  316. skeleton_p_price: firstLoading,
  317. }"
  318. >
  319. <span>
  320. {{
  321. goodsDetail.data.goodsMoney == null
  322. ? "面议"
  323. : goodsDetail.data.goodsMoney
  324. }}
  325. </span>
  326. </strong>
  327. </div>
  328. </div>
  329. </div>
  330. </div>
  331. </div>
  332. <div v-if="
  333. goodsDetail.data.goodsMinOrder != undefined &&
  334. goodsDetail.data.goodsMinOrder
  335. " class="goodsMinOrder">
  336. <span class="">{{ L["最低采购数量:"] }}</span>
  337. <strong>
  338. <span>
  339. {{
  340. goodsDetail.data.goodsMinOrder == null
  341. ? "--"
  342. : goodsDetail.data.goodsMinOrder
  343. }}
  344. </span>
  345. </strong>
  346. </div>
  347. <!-- 商品简介 start -->
  348. <div class="summary-info">
  349. <div
  350. v-if="
  351. goodsDetail.data.goodsSummary != undefined &&
  352. goodsDetail.data.goodsSummary
  353. "
  354. class="summary_html"
  355. :style="goodsDetail.data.goodsSummaryBg"
  356. >
  357. <div
  358. class="summary_htmls"
  359. v-html="goodsDetail.data.goodsSummary"
  360. ></div>
  361. </div>
  362. </div>
  363. <!-- 规格 start -->
  364. <div
  365. class="goods_spec"
  366. v-if="
  367. goodsDetail.data.specs &&
  368. goodsDetail.data.specs.length > 0
  369. "
  370. >
  371. <div
  372. class="goods_spec_pre flex_row_start_start"
  373. v-for="(specItem, specIndex) in goodsDetail.data.specs"
  374. :key="specIndex"
  375. >
  376. <div
  377. class="goods_spec_pre_title"
  378. :title="specItem.specName"
  379. >
  380. {{ specItem.specName }}
  381. </div>
  382. <div class="goods_spec_pre_list flex_row_start_center">
  383. <!-- checkState : 1-选中,2-可选,3-禁用 -->
  384. <!-- 禁止选择 -->
  385. <div
  386. class="specval_pre cursor_pointer"
  387. :class="{
  388. specval_pre_disabled: sepcValItem.checkState == '3',
  389. }"
  390. v-for="(
  391. sepcValItem, specValIndex
  392. ) in specItem.specValueList"
  393. :key="specValIndex"
  394. v-show="sepcValItem.checkState == '3'"
  395. >
  396. <div
  397. class="specval_pre_image"
  398. :style="{
  399. backgroundImage: 'url(' + sepcValItem.image + ')',
  400. }"
  401. :title="sepcValItem.specValue"
  402. v-if="sepcValItem.image"
  403. ></div>
  404. <span class="specval_pre_text" v-else>{{
  405. sepcValItem.specValue
  406. }}</span>
  407. </div>
  408. <!-- 可选择 -->
  409. <div
  410. class="specval_pre cursor_pointer"
  411. :class="{
  412. specval_pre_active: sepcValItem.checkState == '1',
  413. }"
  414. v-for="(
  415. sepcValItem, specValIndex
  416. ) in specItem.specValueList"
  417. :key="specValIndex"
  418. @click="
  419. selectSpecVal(
  420. 'choice',
  421. specItem.specId,
  422. sepcValItem.specValueId
  423. )
  424. "
  425. v-show="sepcValItem.checkState != '3'"
  426. >
  427. <div
  428. class="goods_image"
  429. :style="{
  430. backgroundImage: 'url(' + goodsImgItem + ')',
  431. }"
  432. ></div>
  433. <div
  434. class="specval_pre_image"
  435. :style="{
  436. backgroundImage: 'url(' + sepcValItem.image + ')',
  437. }"
  438. :title="sepcValItem.specValue"
  439. v-if="sepcValItem.image"
  440. ></div>
  441. <span class="specval_pre_text" v-else>{{
  442. sepcValItem.specValue
  443. }}</span>
  444. <img
  445. src="/goods/check_mark.png"
  446. alt=""
  447. class="check_mark"
  448. v-if="sepcValItem.checkState == '1'"
  449. />
  450. </div>
  451. </div>
  452. </div>
  453. </div>
  454. <!-- 规格 end -->
  455. </div>
  456. <!-- 商品未下架即正常商品 end -->
  457. <template v-if="!firstLoading">
  458. <!-- 立即购买 加入购物车 收藏 ,分享 start-->
  459. <!-- 商品已下架 start -->
  460. <div
  461. class="options_btn"
  462. v-if="goodsDetail.data.state && goodsDetail.data.state != 3"
  463. >
  464. <p class="option_desc">{{L['商品已下架,欢迎挑选其他商品~']}}</p>
  465. <div class="goods_off_shelves">{{ L["商品已下架"] }}</div>
  466. <!-- 商品下架时的推荐商品 start -->
  467. <div class="recoOffShop">
  468. <div
  469. class="reCon"
  470. v-for="(
  471. {
  472. goodsImage,
  473. goodsName,
  474. goodsPrice,
  475. defaultProductId,
  476. },
  477. index
  478. ) in recomOffShop.data"
  479. :key="index"
  480. >
  481. <div class="reComImg">
  482. <router-link
  483. :to="'/goods/detail/'+ calcProductName(goodsName) +'_'+ defaultProductId"
  484. target="_blank"
  485. >
  486. <img :src="goodsImage" alt="" />
  487. </router-link>
  488. </div>
  489. <router-link
  490. class="recomName"
  491. :to="'/goods/detail/'+ calcProductName(goodsName) +'_'+ defaultProductId"
  492. target="_blank"
  493. >
  494. {{ goodsName }}
  495. </router-link>
  496. <span class="recomPrice">{{ goodsMoney }}</span>
  497. </div>
  498. </div>
  499. <!-- 商品下架时的推荐商品 end -->
  500. </div>
  501. <!-- 商品已下架 end -->
  502. <!-- 普通(活动)正常商品 start -->
  503. <div class="options_btn flex_row_start_center" v-else>
  504. <div
  505. class="goods_code"
  506. id="qrcodeAct"
  507. v-show="isShowQr"
  508. ></div>
  509. <div v-if="goodsDetail.data.state === 3" class="buy_now flex_row_center_center mt-20" @click="goBuy">
  510. {{ L["发送询盘"] }}
  511. </div>
  512. <div
  513. class="collection_share_btn flex_row_start_start mt-20"
  514. v-if="goodsDetail.data.state == 3"
  515. >
  516. <div
  517. class="collection_btn flex_row_start_center cursor_pointer"
  518. @click="collectGoods"
  519. >
  520. <img
  521. src="/goods/collection.png"
  522. alt=""
  523. v-if="goodsDetail.data.followGoods"
  524. />
  525. <img src="/goods/collection1.png" alt="" v-else />
  526. <span>{{
  527. goodsDetail.data.followGoods ? L["已收藏"] : L["收藏"]
  528. }}</span>
  529. </div>
  530. <!-- <div class="share_btn">
  531. <AddThis publicId="ra-5ab34ca22008ed41" />
  532. </div> -->
  533. </div>
  534. </div>
  535. <!-- 普通(活动)正常商品 end -->
  536. <!-- 立即购买 加入购物车 收藏 ,分享 end-->
  537. </template>
  538. </div>
  539. </div>
  540. <!-- 商品详细信息 end -->
  541. <!-- 相关推荐 start -->
  542. <div class="more_goods" v-if="goodsDetail.data.state == 3">
  543. <div class="more_goods_title">{{ L["看了又看"] }}</div>
  544. <div class="more_goods_list flex_column_center_center">
  545. <template v-if="firstLoading">
  546. <div
  547. class="more_goods_pre"
  548. v-for="(recommendItem, recommendIndex) in [
  549. { a: 1 },
  550. { b: 2 },
  551. { c: 3 },
  552. ]"
  553. :key="recommendIndex"
  554. >
  555. <router-link
  556. target="_blank"
  557. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  558. >
  559. <div
  560. :class="{
  561. more_goods_pre_img: true,
  562. skeleton_more_goods_pre_img: firstLoading,
  563. }"
  564. ></div>
  565. <p
  566. :class="{
  567. skeleton_more_goods_pre_goods_name: firstLoading,
  568. }"
  569. >
  570. {{ recommendItem.goodsName }}
  571. </p>
  572. <p>
  573. <span
  574. :class="{
  575. skeleton_more_goods_pre_goods_price: firstLoading,
  576. }"
  577. ></span>
  578. </p>
  579. </router-link>
  580. </div>
  581. </template>
  582. <template v-else>
  583. <div
  584. class="more_goods_pre"
  585. v-for="(
  586. recommendItem, recommendIndex
  587. ) in recommendeList.data"
  588. :key="recommendIndex"
  589. v-show="recommendIndex < 2"
  590. >
  591. <router-link
  592. target="_blank"
  593. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  594. >
  595. <div class="more_goods_pre_img flex_row_center_center">
  596. <img
  597. :src="recommendItem.goodsImage"
  598. :title="recommendItem.goodsName"
  599. />
  600. </div>
  601. <p>{{ recommendItem.goodsName }}</p>
  602. <p>{{ recommendItem.goodsMoney }}</p>
  603. </router-link>
  604. </div>
  605. </template>
  606. </div>
  607. </div>
  608. <!-- 相关推荐 end -->
  609. </div>
  610. <!-- 商品相关 end -->
  611. <!-- 店铺,及各种信息的切换 start -->
  612. <div
  613. class="container flex_row_start_start"
  614. ref="container"
  615. id="container"
  616. >
  617. <div class="left">
  618. <div class="store_info" v-if="goodsDetail.data.storeInf">
  619. <div class="store_info_title flex_row_start_center">
  620. <span
  621. class="store_type"
  622. v-if="goodsDetail.data.storeInf.isOwnStore == 1"
  623. >{{ L["自营"] }}</span
  624. >
  625. <router-link
  626. target="_blank"
  627. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  628. >
  629. <span class="store_title" @click="goStore()">{{
  630. goodsDetail.data.storeInf.storeName
  631. }}</span>
  632. </router-link>
  633. </div>
  634. <div class="store_des">
  635. <div class="store_des_pre pre_service">
  636. <span>{{ L["客服电话"] }}:</span>
  637. <span>{{ goodsDetail.data.storeInf.servicePhone }}</span>
  638. </div>
  639. <div class="store_des_pre pre_service">
  640. <span>{{ L["主营商品"] }}:</span>
  641. <span>{{ goodsDetail.data.storeInf.mainBusiness }}</span>
  642. </div>
  643. </div>
  644. <div class="store_btn flex_row_center_center">
  645. <div class="store_btn_pre go_store_btn flex_row_center_center">
  646. <!-- <img src="/goods/store.png" alt="" class="btn_pre_img" /> -->
  647. <router-link
  648. target="_blank"
  649. :to="'/store/'+ calcProductName(goodsDetail.data.storeInf.storeName) +'_'+ goodsDetail.data.storeInf.storeId"
  650. >
  651. {{ L["进入店铺"] }}
  652. </router-link>
  653. </div>
  654. <div
  655. class="store_btn_pre follow_btn flex_row_center_center"
  656. @click="focusStore"
  657. >
  658. <!-- <img
  659. src="/goods/collection.png"
  660. alt=""
  661. v-if="goodsDetail.data.storeInf.isFollowStore"
  662. />
  663. <img src="/goods/no_collection.png" alt="" v-else /> -->
  664. <span>{{
  665. goodsDetail.data.storeInf.isFollowStore
  666. ? L["取消关注"]
  667. : L["关注店铺"]
  668. }}</span>
  669. </div>
  670. </div>
  671. </div>
  672. <!-- 店铺推荐及热门收藏 end -->
  673. </div>
  674. <!-- 商品详情 评价 商品服务 店铺热销 start-->
  675. <div class="goods_description">
  676. <div class="goods_description_title flex_row_between_center">
  677. <div class="description_title_left flex_row_start_center">
  678. <span
  679. class="cursor_pointer"
  680. :class="{
  681. description_active: goodsDesctionType == 'detail',
  682. }"
  683. @click="goodsDescType('detail')"
  684. >{{ L["商品详情"] }}</span
  685. >
  686. <span
  687. class="cursor_pointer"
  688. v-if="
  689. goodsDetail.data.goodsAnnexList &&
  690. goodsDetail.data.goodsAnnexList.length > 0
  691. "
  692. :class="{
  693. description_active: goodsDesctionType == 'evaluate',
  694. }"
  695. @click="goodsDescType('evaluate')"
  696. >{{ L["商品说明"] }}</span
  697. >
  698. <span
  699. class="cursor_pointer"
  700. v-if="
  701. goodsDetail.data.serviceLabels &&
  702. goodsDetail.data.serviceLabels.length > 0
  703. "
  704. :class="{
  705. description_active: goodsDesctionType == 'service',
  706. }"
  707. @click="goodsDescType('service')"
  708. >{{ L["商品服务"] }}</span
  709. >
  710. <span
  711. class="cursor_pointer"
  712. :class="{
  713. description_active: goodsDesctionType == 'salestore',
  714. }"
  715. @click="goodsDescType('salestore')"
  716. >{{ L["店铺推荐"] }}</span
  717. >
  718. </div>
  719. </div>
  720. <div class="goods_description_con">
  721. <!-- 商品详情,规格参数,及详情富文本 start-->
  722. <div class="goods_des_con" v-if="goodsDesctionType == 'detail'">
  723. <div
  724. v-if="
  725. goodsDetail.data.brandName ||
  726. (goodsDetail.data.goodsParameterList &&
  727. goodsDetail.data.goodsParameterList.length > 0) ||
  728. goodsDetail.data.goodsDetails
  729. "
  730. >
  731. <div class="brand" v-if="goodsDetail.data.brandName">
  732. <span>{{ L["品牌"] }}: </span>
  733. <span>{{ goodsDetail.data.brandName }}</span>
  734. </div>
  735. <div
  736. v-if="
  737. goodsDetail.data.goodsParameterList &&
  738. goodsDetail.data.goodsParameterList.length > 0
  739. "
  740. >
  741. <div
  742. class="goods_parameter_list"
  743. :class="{ goods_paramter_more: !parameterShow }"
  744. >
  745. <div
  746. class="goods_parameter_pre"
  747. v-for="(parameterItem, paramterIndex) in goodsDetail
  748. .data.goodsParameterList"
  749. :key="paramterIndex"
  750. >
  751. <span>{{ parameterItem.parameterName }}: </span>
  752. <span>{{ parameterItem.parameterValue }}</span>
  753. </div>
  754. </div>
  755. <div
  756. class="collapse_unfold flex_row_center_center cursor_pointer"
  757. v-if="goodsDetail.data.goodsParameterList.length > 16"
  758. @click="openParameter"
  759. >
  760. <span>{{
  761. !parameterShow ? L["查看全部"] : L["收起全部"]
  762. }}</span>
  763. <i
  764. class="iconfont icon-ziyuan11-copy"
  765. v-if="!parameterShow"
  766. ></i>
  767. <i class="iconfont icon-ziyuan11-copy-copy" v-else></i>
  768. </div>
  769. </div>
  770. <div
  771. v-if="
  772. goodsDetail.data.topTemplateContent != undefined &&
  773. goodsDetail.data.topTemplateContent
  774. "
  775. class="goods_html"
  776. :style="goodsDetail.data.topTemplateContentBg"
  777. >
  778. <div
  779. class="goods_htmls"
  780. v-html="goodsDetail.data.topTemplateContent"
  781. ></div>
  782. </div>
  783. <div
  784. v-if="
  785. goodsDetail.data.goodsDetails != undefined &&
  786. goodsDetail.data.goodsDetails
  787. "
  788. class="goods_html"
  789. :style="goodsDetail.data.goodsDetailsBg"
  790. >
  791. <div
  792. class="goods_htmls"
  793. v-html="goodsDetail.data.goodsDetails"
  794. ></div>
  795. </div>
  796. <div
  797. v-if="
  798. goodsDetail.data.bottomTemplateContent != undefined &&
  799. goodsDetail.data.bottomTemplateContent
  800. "
  801. class="goods_html"
  802. :style="goodsDetail.data.bottomTemplateContentBg"
  803. >
  804. <div
  805. class="goods_htmls"
  806. v-html="goodsDetail.data.bottomTemplateContent"
  807. ></div>
  808. </div>
  809. </div>
  810. <div v-else>
  811. <SldCommonEmpty
  812. :tip="L['该商品暂无详情~']"
  813. totalWidth="934px"
  814. />
  815. </div>
  816. </div>
  817. <!-- 商品详情,规格参数,及详情富文本 end-->
  818. <!-- 商品评价 start -->
  819. <div
  820. class="goods_comments"
  821. v-if="goodsDesctionType == 'evaluate'"
  822. >
  823. <!--说明书下载begin-->
  824. <div
  825. class="download_warp"
  826. v-if="
  827. goodsDetail.data.goodsAnnexList &&
  828. goodsDetail.data.goodsAnnexList.length > 0
  829. "
  830. >
  831. <span style="margin: 10px; font-size: 14px"
  832. >{{ L["说明书下载"] }}:
  833. </span>
  834. <br />
  835. <p
  836. style="margin: 10px; font-size: 14px"
  837. v-for="(item, index) in goodsDetail.data.goodsAnnexList"
  838. :key="index"
  839. >
  840. <a
  841. style="text-decoration: underline"
  842. href="javascript:;"
  843. @click="downloadAdd(item)"
  844. >{{ item.annexName }}</a
  845. >
  846. </p>
  847. </div>
  848. <div v-else>
  849. {{ L["暂无说明"] }}
  850. </div>
  851. <!--说明书下载end-->
  852. </div>
  853. <!-- 商品评价 end -->
  854. <!-- 商品服务 start -->
  855. <div
  856. class="goods_server_list"
  857. v-if="goodsDesctionType == 'service'"
  858. >
  859. <div
  860. v-if="
  861. goodsDetail.data.serviceLabels &&
  862. goodsDetail.data.serviceLabels.length > 0
  863. "
  864. >
  865. <div
  866. class="goods_server_pre"
  867. v-for="(serverItem, serverIndex) in goodsDetail.data
  868. .serviceLabels"
  869. :key="serverIndex"
  870. >
  871. <div class="server_pre_top flex_row_start_center">
  872. <span class="server_pre_tips"></span>
  873. <span class="server_pre_name">{{
  874. serverItem.labelName
  875. }}</span>
  876. </div>
  877. <div class="server_pre_content">
  878. {{ serverItem.description }}
  879. </div>
  880. </div>
  881. </div>
  882. <div v-else>
  883. <SldCommonEmpty
  884. :tip="L['暂无商品服务~']"
  885. totalWidth="934px"
  886. />
  887. </div>
  888. </div>
  889. <!-- 商品服务 end -->
  890. <!-- 店铺推荐 start -->
  891. <div
  892. class="store_hot_sales"
  893. v-if="goodsDesctionType == 'salestore'"
  894. >
  895. <div
  896. v-if="
  897. recommendedList.data && recommendedList.data.length > 0
  898. "
  899. >
  900. <div class="store_hot_sales_list">
  901. <div
  902. class="goods_pre flex_column_between_start"
  903. v-for="(
  904. recommendItem, recommendIndex
  905. ) in recommendedList.data"
  906. :key="recommendIndex"
  907. >
  908. <router-link
  909. target="_blank"
  910. :to="'/goods/detail/'+ calcProductName(recommendItem.goodsName) +'_'+ recommendItem.defaultProductId"
  911. >
  912. <div class="flex_column_start_start">
  913. <div
  914. class="goods_pre_img"
  915. :style="{
  916. backgroundImage:
  917. 'url(' + recommendItem.goodsImage + ')',
  918. }"
  919. ></div>
  920. <div class="goods_name">
  921. {{ recommendItem.goodsName }}
  922. </div>
  923. </div>
  924. <div class="goods_price">
  925. <div class="selling_price">
  926. {{ recommendItem.goodsMoney }}
  927. </div>
  928. </div>
  929. </router-link>
  930. </div>
  931. </div>
  932. <div
  933. class="flex_row_end_center sld_pagination sld_page_bottom"
  934. v-if="
  935. recommendeData.data.pagination &&
  936. recommendeData.data.pagination.total
  937. "
  938. >
  939. <el-pagination
  940. @current-change="handleCurrentChangeSales"
  941. v-model:currentPage="salesCurrent"
  942. :page-size="salesPageSize"
  943. layout="prev, pager, next, jumper"
  944. :total="recommendeData.data.pagination.total"
  945. :hide-on-single-page="true"
  946. >
  947. </el-pagination>
  948. </div>
  949. </div>
  950. <div
  951. v-if="
  952. recommendedList.data && recommendedList.data.length == 0
  953. "
  954. class="flex_column_center_center empty_data"
  955. >
  956. <SldCommonEmpty
  957. :tip="L['暂无相关商品~']"
  958. totalWidth="934px"
  959. />
  960. </div>
  961. </div>
  962. <!-- 店铺推荐 end -->
  963. </div>
  964. </div>
  965. <!-- 商品详情 评价 商品服务 店铺热销 end-->
  966. </div>
  967. <!-- 店铺,及各种信息的切换 end -->
  968. </div>
  969. <!-- 商品主要信息 end -->
  970. </div>
  971. <EnquiryModal
  972. v-if="enquiryVis"
  973. :itemType="'GOODS'"
  974. :itemId="productId"
  975. @closeLoingModal="closeEnquiryModal"
  976. />
  977. </div>
  978. </div>
  979. </template>
  980. <script setup>
  981. import addrData from "@/assets/area.json";
  982. import { ElMessage, ElRate, ElDialog, ElPagination } from "element-plus";
  983. import { qrcanvas } from "qrcanvas";
  984. // import { lang_zn } from "@/assets/language/zh";
  985. import { getCurLanguage } from '@/composables/common.js';
  986. import { goodsInfo, useUserInfo } from "@/store/user.js";
  987. import { useFiltersStore } from "@/store/filter.js";
  988. const filtersStore = useFiltersStore();
  989. const goodsInfox = goodsInfo();
  990. const configInfo = useUserInfo();
  991. // const L = lang_zn;
  992. const L = getCurLanguage();
  993. const firstLoading = ref(true); //是否第一次加载
  994. const router = useRouter();
  995. const route = useRoute();
  996. const store = ref();
  997. const imgVisible = ref(false);
  998. const imgSource = ref("");
  999. const imgIndex = ref(-1);
  1000. const vid = ref(0); //店铺id
  1001. const proxy = getCurrentInstance();
  1002. const goodsDetail = reactive({ data: {} }); //商品详情数据
  1003. const isChoice = ref("default"); //是默认选中的,还是点击选择规格之后的 default:默认 choice:选择
  1004. const productId = ref(""); //货品id
  1005. const fullDisList = reactive({ data: [] }); //满优惠促销列表
  1006. const couponList = reactive({ data: [] }); //获取店铺优惠券列表
  1007. const pictureBig = ref(null); //大图的信息
  1008. const maskBox = ref(null); //遮罩盒子的信息
  1009. const maskShow = ref(false); //遮罩是否显示
  1010. const previewBox = ref(null); // 左侧主图元素信息
  1011. const zoomBox = ref(null); //左侧主图的父元素的信息
  1012. const defaultImage = ref(""); //默认主图路径
  1013. const currentDefaultImage = ref("0"); //默认主图显示第一张
  1014. const currentSpecNum = ref(1); //商品编辑数量,默认数量为1
  1015. const recommendeList = reactive({ data: [] }); //看了又看商品
  1016. const storePopularList = reactive({ data: [] }); //店铺推荐及热门收藏
  1017. const storePopularType = ref("collection"); //店铺推荐及热门收藏,默认显示店铺推荐
  1018. const goodsDesctionType = ref("detail"); //商品详情,评价,商品服务,店铺热销,默认显示商品详情
  1019. const goodsCommentsInfo = reactive({ data: {} }); //商品评价信息
  1020. const evaluationType = ref(""); //商品评价类型,默认显示全部
  1021. const recommendedList = reactive({ data: [] }); //店铺推荐列表
  1022. const evaluationCurrent = ref(1); //评价列表默认第一页
  1023. const evaluationPageSize = ref(5); //评价列表默认一页显示5条数据
  1024. const salesCurrent = ref(1); //店铺热销列表默认一页
  1025. const salesPageSize = ref(20); //店铺热销列表默认一页显示20条数据
  1026. const recommendeData = reactive({ data: {} }); //店铺热销推荐数据
  1027. const couponModel = ref(false); //优惠券弹框是否显示
  1028. const cartListData = reactive({ data: goodsInfox.cartListData }); //获取vux的store中的购物车数据
  1029. const enquiryVis = ref(false); //登录弹框是否显示,默认不显示
  1030. const container = ref(null); // 商品详情页底部内容区
  1031. const containerTop = ref(0); //商品详情页底部内容区的top值
  1032. const containCon = ref(false); //固定内容区头部
  1033. const fullDiscountModel = ref(false); //满优惠弹框是否显示
  1034. const goodsPictureList = ref(null); //商品图片列表
  1035. const curCouponPage = ref(1); //当前为第一页优惠券
  1036. const couponPageSize = ref(6); //优惠券默认一页显示6条数据
  1037. const wxShareCode = ref(false); //微信分享二维码是否显示
  1038. const parameterShow = ref(false); //规格参数查看是否查看全部,默认为否
  1039. const score = ref(0); //好评率
  1040. const colors = ref(["#E2231A", "#E2231A", "#E2231A"]); //星星颜色
  1041. const curAddr = ref(-1);
  1042. const curAddrName = ref("");
  1043. const addrIdx = ref(0);
  1044. const otherAddrIdx = ref(0);
  1045. const othTopIdx = reactive({
  1046. 0: 0,
  1047. 1: 0,
  1048. 2: 0,
  1049. });
  1050. const otherTree = ref([
  1051. addrData[othTopIdx["0"]],
  1052. addrData[othTopIdx["0"]].children[othTopIdx["1"]],
  1053. ]);
  1054. const othAddrDe = ref(addrData);
  1055. const addrDialogVisible = ref(false);
  1056. const logFlag = ref(configInfo.loginFlag);
  1057. // 促销活动信息
  1058. const preSellInfo = reactive({ data: {} });
  1059. const pinInfo = reactive({ data: {} });
  1060. const seckillInfo = reactive({ data: {} });
  1061. const ladderInfo = reactive({ data: {} });
  1062. const address_list = reactive({ data: [] });
  1063. const isShowQr = ref(false);
  1064. const secInt = ref("");
  1065. const time = reactive({
  1066. day: "00",
  1067. hours: "00",
  1068. minutes: "00",
  1069. seconds: "00",
  1070. });
  1071. const judgeStock = computed(() => {
  1072. return (
  1073. goodsDetail.data.defaultProduct.productStock == 0 ||
  1074. (JSON.stringify(preSellInfo.data) != "{}" &&
  1075. preSellInfo.data.presellStock == 0) ||
  1076. (JSON.stringify(pinInfo.data) != "{}" && pinInfo.data.spellStock == 0) ||
  1077. (JSON.stringify(seckillInfo.data) != "{}" &&
  1078. seckillInfo.data.seckillStock == 0)
  1079. );
  1080. });
  1081. // 促销活动信息end
  1082. const scrollHandle = async (e) => {
  1083. if (process.client) {
  1084. let elementScrollTop = e.srcElement.scrollingElement.scrollTop; //获取页面滚动高度
  1085. if (
  1086. document.getElementById("container") &&
  1087. elementScrollTop > document.getElementById("container").offsetTop
  1088. ) {
  1089. containCon.value = true;
  1090. await proxy.$nextTick();
  1091. } else {
  1092. containCon.value = false;
  1093. }
  1094. }
  1095. };
  1096. // 点击播放视频
  1097. const playV = () => {
  1098. currentDefaultImage.value = -1;
  1099. defaultImage.value = "";
  1100. maskShow.value = false;
  1101. videoEnd();
  1102. };
  1103. //获取商品详情数据
  1104. const getInitDataStatic = async (proId) => {
  1105. let params = {
  1106. productId: proId,
  1107. };
  1108. const { data: value, pending: pending } = await useFetchRaw(
  1109. apiUrl + "v3/goods/front/goods/details",
  1110. { params: params }
  1111. );
  1112. const res = value._rawValue;
  1113. if (res.state == 200) {
  1114. let staticData = [
  1115. "brandId",
  1116. "brandName",
  1117. "categoryPath",
  1118. "categoryIdPath",
  1119. "goodsBrief",
  1120. "goodsDetails",
  1121. "goodsSummary",
  1122. "goodsBrief",
  1123. "goodsId",
  1124. "goodsName",
  1125. "goodsParameterList",
  1126. "goodsVideo",
  1127. "topTemplateContent",
  1128. "bottomTemplateContent",
  1129. "goodsAnnexList",
  1130. "serviceLabels",
  1131. "goodsMoney",
  1132. "goodsMinOrder",
  1133. "goodsParameterList",
  1134. ];
  1135. staticData.forEach((item) => {
  1136. if (item == "categoryPath") {
  1137. goodsDetail.data.categoryPath = res.data.categoryPath.split("->");
  1138. } else if (item == "categoryIdPath") {
  1139. goodsDetail.data.categoryIdPath = [
  1140. res.data.categoryId1,
  1141. res.data.categoryId2,
  1142. res.data.categoryId3,
  1143. ];
  1144. } else {
  1145. goodsDetail.data[item] = res.data[item];
  1146. }
  1147. });
  1148. if (goodsDetail.data.goodsSummary) {
  1149. goodsDetail.data.goodsSummary = quillEscapeToHtml(
  1150. goodsDetail.data.goodsSummary
  1151. );
  1152. //处理背景样式
  1153. if (
  1154. goodsDetail.data.goodsSummary.indexOf(
  1155. '<p style="display:none;" data-background="'
  1156. ) != -1
  1157. ) {
  1158. let bg = goodsDetail.data.goodsSummary
  1159. .split('<p style="display:none;" data-background="')[1]
  1160. .split('">')[0];
  1161. goodsDetail.data.goodsSummaryBg = bg;
  1162. }
  1163. }
  1164. if (goodsDetail.data.topTemplateContent) {
  1165. goodsDetail.data.topTemplateContent = quillEscapeToHtml(
  1166. goodsDetail.data.topTemplateContent
  1167. );
  1168. //处理背景样式
  1169. if (
  1170. goodsDetail.data.topTemplateContent.indexOf(
  1171. '<p style="display:none;" data-background="'
  1172. ) != -1
  1173. ) {
  1174. let bg = goodsDetail.data.topTemplateContent
  1175. .split('<p style="display:none;" data-background="')[1]
  1176. .split('">')[0];
  1177. goodsDetail.data.topTemplateContentBg = bg;
  1178. }
  1179. }
  1180. if (goodsDetail.data.goodsDetails) {
  1181. goodsDetail.data.goodsDetails = quillEscapeToHtml(
  1182. goodsDetail.data.goodsDetails
  1183. );
  1184. //处理背景样式
  1185. if (
  1186. goodsDetail.data.goodsDetails.indexOf(
  1187. '<p style="display:none;" data-background="'
  1188. ) != -1
  1189. ) {
  1190. let bg = goodsDetail.data.goodsDetails
  1191. .split('<p style="display:none;" data-background="')[1]
  1192. .split('">')[0];
  1193. goodsDetail.data.goodsDetailsBg = bg;
  1194. }
  1195. }
  1196. if (goodsDetail.data.bottomTemplateContent) {
  1197. goodsDetail.data.bottomTemplateContent = quillEscapeToHtml(
  1198. goodsDetail.data.bottomTemplateContent
  1199. );
  1200. //处理背景样式
  1201. if (
  1202. goodsDetail.data.bottomTemplateContent.indexOf(
  1203. '<p style="display:none;" data-background="'
  1204. ) != -1
  1205. ) {
  1206. let bg = goodsDetail.data.bottomTemplateContent
  1207. .split('<p style="display:none;" data-background="')[1]
  1208. .split('">')[0];
  1209. goodsDetail.data.bottomTemplateContentBg = bg;
  1210. }
  1211. }
  1212. currentDefaultImage.value = 0;
  1213. vid.value = res.data.storeInf.storeId;
  1214. // setTimeout(() => {
  1215. // sldStatEvent({ behaviorType: 'gpv', goodsId: goodsDetail.data.goodsId, storeId: vid.value });
  1216. // }, 3000)
  1217. } else {
  1218. ElMessage.error(res.msg);
  1219. }
  1220. if (!pending._rawValue) {
  1221. getInitDataDynamic(productId.value);
  1222. getRecommend();
  1223. getStorePopular();
  1224. getEvaluation();
  1225. addLog();
  1226. }
  1227. };
  1228. getInitDataStatic(calcProductId(route.path));
  1229. onMounted(() => {
  1230. setTimeout(() => {
  1231. sldStatEvent({ behaviorType: 'gpv', goodsId: goodsDetail.data.goodsId, storeId: vid.value,pageUrl: defaultUrl + router.currentRoute.value.path, referrerPageUrl: apiUrl });
  1232. }, 3000)
  1233. });
  1234. productId.value = calcProductId(route.path);
  1235. if (!logFlag.value) {
  1236. addrIdx.value = 1;
  1237. }
  1238. const getInitDataDynamic = async (proId, updateType) => {
  1239. let params = {
  1240. productId: proId,
  1241. };
  1242. const { data: value, pending: pending } = await useFetchRaw(
  1243. apiUrl + "v3/goods/front/goods/details2",
  1244. { params: params,
  1245. headers:{Authorization:'Bearer ' + filtersStore.getToken}
  1246. }
  1247. );
  1248. const res = value._rawValue;
  1249. if (res.state == 200) {
  1250. useHead({
  1251. title: res.data.seoInfo.seoTitle || 'Goods Detail',
  1252. meta: [
  1253. {
  1254. name: "description",
  1255. content: res.data.seoInfo.seoDesc,
  1256. },
  1257. {
  1258. name: "keywords",
  1259. content: res.data.seoInfo.seoKeywords,
  1260. },
  1261. ],
  1262. });
  1263. defaultImage.value = res.data.defaultProduct.goodsPics[0];
  1264. let dynamicData = [
  1265. "defaultProduct",
  1266. "deliverInfo",
  1267. "effectSpecValueIds",
  1268. "followGoods",
  1269. "specs",
  1270. "storeInf",
  1271. "sales",
  1272. "state",
  1273. "shareLink",
  1274. "shareImage",
  1275. "goodsMinOrder",
  1276. "isVirtualGoods",
  1277. ];
  1278. dynamicData.forEach((item) => {
  1279. goodsDetail.data[item] = res.data[item];
  1280. });
  1281. if (goodsDetail.data.state != 3) {
  1282. getRecom();
  1283. }
  1284. if (goodsDetail.data.defaultProduct.promotionType == 103) {
  1285. getPreSell(res.data.defaultProduct.productId);
  1286. } else if (goodsDetail.data.defaultProduct.promotionType == 102) {
  1287. getPin();
  1288. } else if (goodsDetail.data.defaultProduct.promotionType == 104) {
  1289. getSeckill();
  1290. } else if (goodsDetail.data.defaultProduct.promotionType == 105) {
  1291. getLadder();
  1292. } else {
  1293. preSellInfo.data = {};
  1294. pinInfo.data = {};
  1295. seckillInfo.data = {};
  1296. ladderInfo.data = {};
  1297. }
  1298. firstLoading.value = false;
  1299. }
  1300. };
  1301. //视频播放结束时触发
  1302. const videoEnd = () => {
  1303. nextTick(() => {
  1304. proxy.refs.video.onended = () => {
  1305. currentDefaultImage.value = 0;
  1306. defaultImage.value = goodsDetail.data.defaultProduct.goodsPics[0];
  1307. };
  1308. });
  1309. };
  1310. // 促销活动信息
  1311. const getPreSell = async (productId) => {
  1312. let param = {
  1313. productId: productId,
  1314. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1315. };
  1316. const { data: value } = await useFetchRaw(
  1317. apiUrl + "v3/promotion/front/preSell/detail",
  1318. { params: param }
  1319. );
  1320. const res = value.value._rawValue;
  1321. if (res.state == 200) {
  1322. let now = new Date();
  1323. let preStartDate = new Date(res.data.startTime);
  1324. let preEndDate = new Date(res.data.endTime);
  1325. preSellInfo.data = res.data;
  1326. let countTime = 0;
  1327. preSellInfo.data.endTime = formatPreTime(new Date(res.data.endTime));
  1328. preSellInfo.data.startTime = formatPreTime(new Date(res.data.startTime));
  1329. if (now > preStartDate && now < preEndDate) {
  1330. preSellInfo.data.pre_run = 2; //活动进行中
  1331. countTime = res.data.distanceEndTime;
  1332. countDown(countTime);
  1333. } else if (now < preStartDate) {
  1334. preSellInfo.data.pre_run = 1; //活动未开始
  1335. countTime =
  1336. (new Date(res.data.startTime).getTime() - now.getTime()) / 1000;
  1337. countDown(countTime);
  1338. } else if (now > preEndDate) {
  1339. preSellInfo.data.pre_run = 3; //活动已结束
  1340. }
  1341. genQrcode();
  1342. } else {
  1343. ElMessage.error(res.msg);
  1344. }
  1345. };
  1346. const getPin = () => {
  1347. let param = {
  1348. productId: goodsDetail.data.defaultProduct.productId,
  1349. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1350. };
  1351. get("v3/promotion/front/spell/detail", param).then((res) => {
  1352. if (res.state == 200) {
  1353. pinInfo.data = res.data;
  1354. let countTime = 0;
  1355. let now = new Date();
  1356. let startTime = new Date(res.data.startTime);
  1357. if (now < startTime) {
  1358. countTime = (startTime.getTime() - now.getTime()) / 1000;
  1359. countDown(countTime);
  1360. } else {
  1361. countTime = res.data.distanceEndTime;
  1362. countDown(countTime);
  1363. }
  1364. genQrcode();
  1365. } else {
  1366. ElMessage.error(res.msg);
  1367. }
  1368. });
  1369. };
  1370. const getSeckill = () => {
  1371. let param = {
  1372. productId: goodsDetail.data.defaultProduct.productId,
  1373. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1374. };
  1375. get("v3/promotion/front/seckill/detail", param).then((res) => {
  1376. if (res.state == 200) {
  1377. seckillInfo.data = res.data;
  1378. let now = new Date();
  1379. let countTime = 0;
  1380. let startTime = new Date(res.data.startTime);
  1381. if (seckillInfo.data.state == 1 || seckillInfo.data.state == 2) {
  1382. countTime = res.data.distanceEndTime;
  1383. countDown(countTime);
  1384. } else {
  1385. countTime = startTime.getTime() - now.getTime();
  1386. }
  1387. genQrcode();
  1388. }
  1389. });
  1390. };
  1391. const countDown = (countTime) => {
  1392. secInt.value = setInterval(() => {
  1393. if (countTime == 0) {
  1394. getInitDataDynamic(calcProductId(route.path));
  1395. clearInterval(secInt.value);
  1396. } else {
  1397. countTime--;
  1398. let day = parseInt(countTime / 60 / 60 / 24);
  1399. let hours = parseInt((countTime / 60 / 60) % 24);
  1400. let minutes = parseInt((countTime / 60) % 60);
  1401. let seconds = parseInt(countTime % 60);
  1402. time.day = day;
  1403. time.hours = hours > 9 ? hours : "0" + hours;
  1404. time.minutes = minutes > 9 ? minutes : "0" + minutes;
  1405. time.seconds = seconds > 9 ? seconds : "0" + seconds;
  1406. }
  1407. }, 1000);
  1408. };
  1409. const formatPreTime = (time) => {
  1410. let op = new Date(time);
  1411. let year = op.getFullYear();
  1412. let month = op.getMonth() + 1;
  1413. let day = op.getDate();
  1414. let hour = op.getHours();
  1415. let minute = op.getMinutes();
  1416. let part1 = [year, month, day]
  1417. .map((i) => (i.toString().length < 2 ? `0${i}` : i))
  1418. .join("-");
  1419. let part2 = [hour, minute]
  1420. .map((i) => (i.toString().length < 2 ? `0${i}` : i))
  1421. .join(":");
  1422. return part1 + " " + part2;
  1423. };
  1424. const getLadder = () => {
  1425. let param = {
  1426. productId: goodsDetail.data.defaultProduct.productId,
  1427. promotionId: goodsDetail.data.defaultProduct.promotionId,
  1428. };
  1429. get("v3/promotion/front/ladder/group/detail", param).then((res) => {
  1430. if (res.state == 200) {
  1431. ladderInfo.data = res.data;
  1432. let now = new Date();
  1433. let countTime = 0;
  1434. let startTime = new Date(res.data.startTime);
  1435. if (now < startTime) {
  1436. countTime = (startTime.getTime() - now.getTime()) / 1000;
  1437. countDown(countTime);
  1438. ladderInfo.data.state = 1;
  1439. } else {
  1440. countTime = res.data.distanceEndTime;
  1441. countDown(countTime);
  1442. ladderInfo.data.state = 2;
  1443. }
  1444. genQrcode();
  1445. }
  1446. });
  1447. };
  1448. const genQrcode = () => {
  1449. if (judgeStock.value) {
  1450. return;
  1451. }
  1452. proxy.$nextTick(() => {
  1453. let canvas = qrcanvas({
  1454. data: goodsDetail.data.shareLink, //二维码内容
  1455. size: 125,
  1456. colorDark: "red",
  1457. });
  1458. setTimeout(() => {
  1459. document.getElementById("qrcodeAct").innerHTML = "";
  1460. document.getElementById("qrcodeAct").appendChild(canvas);
  1461. }, 10);
  1462. });
  1463. };
  1464. // 促销活动信息end
  1465. const recomOffShop = reactive({ data: [] });
  1466. const getRecom = () => {
  1467. get("v3/goods/front/goods/goodsList", {
  1468. storeId: goodsDetail.data.categoryId1,
  1469. }).then((res) => {
  1470. if (res.state == 200) {
  1471. let top = Math.floor(Math.random() * (res.data.list.length - 8)) + 8;
  1472. let end = top - 8;
  1473. recomOffShop.data = res.data.list
  1474. .filter(
  1475. (item) =>
  1476. item.defaultProductId != goodsDetail.data.defaultProduct.productId
  1477. )
  1478. .slice(end, top);
  1479. }
  1480. });
  1481. };
  1482. //添加足迹
  1483. const addLog = () => {
  1484. let params = {
  1485. productId: productId.value,
  1486. };
  1487. post("v3/member/front/productLookLog/add", params).then((res) => {});
  1488. };
  1489. //记录下载
  1490. const downloadAdd = (item) => {
  1491. let url = item.annexUrl;
  1492. let name = item.annexName;
  1493. if (filtersStore.getLoginFlag) {
  1494. post("v3/member/front/download/add", {
  1495. goodsId: item.goodsId,
  1496. annexId: item.annexId,
  1497. }).then((res) => {
  1498. });
  1499. }
  1500. const link = document.createElement("a");
  1501. fetch(url)
  1502. .then((res) => res.blob())
  1503. .then((blob) => {
  1504. link.href = URL.createObjectURL(blob);
  1505. link.download = name;
  1506. document.body.appendChild(link);
  1507. link.click();
  1508. window.URL.revokeObjectURL(link.href);
  1509. document.body.removeChild(link);
  1510. });
  1511. };
  1512. /**
  1513. * 选择规格值
  1514. * @param type:类型 值:choice,规格选择 default:默认
  1515. * @param specId:父级规格值
  1516. * @param specValueId:点击的当前的规格值
  1517. */
  1518. const selectSpecVal = (type, specId, specValueId) => {
  1519. isChoice.value = type == "choice" ? "choice" : "default";
  1520. let curParSpec = []; //当前点击的规格的父级id的当前项
  1521. curParSpec = goodsDetail.data.specs.filter((item) => item.specId == specId);
  1522. let curSPec = []; //当前点击的规格的规格id的当前项
  1523. curSPec = curParSpec[0].specValueList.filter(
  1524. (item1) => item1.specValueId == specValueId
  1525. );
  1526. curSPec[0].checkState = 1;
  1527. //被选择的规格值的id
  1528. let choiceSpecIds = [];
  1529. goodsDetail.data.specs.forEach((item) => {
  1530. if (item.specId != specId) {
  1531. item.specValueList.forEach((item1) => {
  1532. if (item1.checkState == "1") {
  1533. // checkState: 1-选中,2-可选,3-禁用
  1534. choiceSpecIds.push(item1.specValueId);
  1535. }
  1536. });
  1537. } else {
  1538. choiceSpecIds.push(specValueId);
  1539. }
  1540. });
  1541. let params = {
  1542. goodsId: goodsDetail.data.goodsId,
  1543. specValueIds: choiceSpecIds.join(","),
  1544. };
  1545. get("v3/goods/front/goods/productInfo", params).then((res) => {
  1546. if (res.state == 200) {
  1547. let result = res.data;
  1548. goodsDetail.data.defaultProduct = result.defaultProduct;
  1549. productId.value = result.defaultProduct.productId;
  1550. goodsDetail.data.specs = result.specs;
  1551. defaultImage.value = goodsDetail.data.defaultProduct.goodsPics[0];
  1552. currentDefaultImage.value = 0;
  1553. isShowQr.value = false;
  1554. // getInitDataDynamic(productId.value)
  1555. }
  1556. });
  1557. };
  1558. //改变数量按钮样式
  1559. const disStyle = reactive({
  1560. //目的是进入商品详情页面就让减按钮呈现禁止状态
  1561. color: "#DDDDDD",
  1562. backgroundColor: "#F8F8F8",
  1563. });
  1564. watch(
  1565. () => currentSpecNum.value,
  1566. () => {
  1567. //监听数量对加和减的样式做出调整
  1568. let productStock = goodsDetail.data.defaultProduct.productStock;
  1569. if (goodsDetail.data.defaultProduct.productStock == 0) {
  1570. productStock = 999;
  1571. }
  1572. if (currentSpecNum.value >= productStock) {
  1573. proxy.refs.add.style.color = "#DDDDDD";
  1574. proxy.refs.add.style.backgroundColor = "#F8F8F8";
  1575. } else if (currentSpecNum.value <= 1) {
  1576. disStyle.color = "#DDDDDD";
  1577. disStyle.backgroundColor = "#F8F8F8";
  1578. } else {
  1579. disStyle.color = "";
  1580. disStyle.backgroundColor = "";
  1581. proxy.refs.add.style.color = "";
  1582. proxy.refs.add.style.backgroundColor = "";
  1583. }
  1584. }
  1585. );
  1586. watch(currentSpecNum, () => {
  1587. if (currentSpecNum.value > goodsDetail.data.defaultProduct.productStock) {
  1588. currentSpecNum.value = goodsDetail.data.defaultProduct.productStock;
  1589. }
  1590. let reg = /\./g;
  1591. let reg0 = /0+\d/;
  1592. if (
  1593. currentSpecNum.value &&
  1594. (reg.test(currentSpecNum.value) || currentSpecNum.value <= 0)
  1595. ) {
  1596. currentSpecNum.value = 1;
  1597. }
  1598. });
  1599. //发送询盘
  1600. const goBuy = () => {
  1601. enquiryVis.value = true;
  1602. };
  1603. //关闭登录弹框
  1604. const closeEnquiryModal = () => {
  1605. enquiryVis.value = false;
  1606. };
  1607. //获取看了又看商品(人气数)
  1608. const getRecommend = async () => {
  1609. let params = {
  1610. storeId: vid.value,
  1611. sort: 5,
  1612. pageSize: 3,
  1613. current: 1,
  1614. };
  1615. const { data: value } = await useFetchRaw(
  1616. apiUrl + "v3/goods/front/goods/goodsList",
  1617. { params: params, key: params.sort.toString() }
  1618. );
  1619. const res = value._rawValue;
  1620. if (res.state == 200) {
  1621. let result = res.data;
  1622. recommendeList.data = result.list;
  1623. recommendeList.data.map(
  1624. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1625. );
  1626. } else {
  1627. ElMessage.error(res.msg);
  1628. }
  1629. };
  1630. //获取店铺推荐的商品(销量数);获取热门收藏的商品(收藏数) type:recommend销量数 collection:收藏数
  1631. const getStorePopular = async (type) => {
  1632. if (type == "" || !type) {
  1633. type = "recommend";
  1634. } else {
  1635. storePopularType.value = type;
  1636. }
  1637. let params = {
  1638. storeId: vid.value,
  1639. sort:
  1640. storePopularType.value == "recommend"
  1641. ? 1
  1642. : storePopularType.value == "collection"
  1643. ? 6
  1644. : "",
  1645. pageSize: 6,
  1646. current: 1,
  1647. };
  1648. const { data: value } = await useFetchRaw(
  1649. apiUrl + "v3/goods/front/goods/goodsList",
  1650. { params: params, key: params.sort.toString() }
  1651. );
  1652. const res = value._rawValue;
  1653. if (res.state == 200) {
  1654. let result = res.data;
  1655. storePopularList.data = result.list;
  1656. storePopularList.data.map(
  1657. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1658. );
  1659. } else {
  1660. ElMessage.error(res.msg);
  1661. }
  1662. };
  1663. //切换商品详情,评价,服务,热销
  1664. const goodsDescType = (type) => {
  1665. // let elementScrollTop =
  1666. // window.pageYOffset ||
  1667. // document.documentElement.scrollTop ||
  1668. // document.body.scrollTop; //获取页面滚动高度
  1669. // if (elementScrollTop > containerTop.value + 33 + 48) {
  1670. // window.scrollTo({
  1671. // top: containerTop.value,
  1672. // behavior: "smooth",
  1673. // });
  1674. // }
  1675. if (type == "" || !type) {
  1676. goodsDesctionType.value = "detail";
  1677. } else {
  1678. goodsDesctionType.value = type;
  1679. }
  1680. if (type == "salestore") {
  1681. getSalestore();
  1682. }
  1683. };
  1684. //获取商品评价
  1685. const getEvaluation = (type) => {
  1686. if (!type) {
  1687. evaluationType.value = "";
  1688. } else {
  1689. evaluationType.value = type;
  1690. }
  1691. let params = {
  1692. productId: productId.value,
  1693. current: evaluationCurrent.value,
  1694. pageSize: evaluationPageSize.value,
  1695. type: evaluationType.value,
  1696. };
  1697. get("v3/goods/front/goods/comment", params).then((res) => {
  1698. if (res.state == 200) {
  1699. let result = res.data;
  1700. goodsCommentsInfo.data = result;
  1701. score.value = Number(goodsCommentsInfo.data.avgScore);
  1702. goodsCommentsInfo.data &&
  1703. goodsCommentsInfo.data.list &&
  1704. goodsCommentsInfo.data.list.length > 0 &&
  1705. goodsCommentsInfo.data.list.map((commentsItem) => {
  1706. commentsItem.memberName =
  1707. commentsItem.memberName.slice(0, 1) +
  1708. "***" +
  1709. commentsItem.memberName.slice(
  1710. commentsItem.memberName.length - 1,
  1711. commentsItem.memberName.length
  1712. );
  1713. });
  1714. } else {
  1715. ElMessage.error(res.msg);
  1716. }
  1717. });
  1718. };
  1719. //评价列表上一页
  1720. const handlePrevCilickChange = () => {
  1721. if (evaluationCurrent.value == 1) {
  1722. evaluationCurrent.value = 1;
  1723. } else {
  1724. evaluationCurrent.value--;
  1725. }
  1726. getEvaluation(evaluationType.value);
  1727. };
  1728. //评价列表下一页
  1729. const handleNextCilickChange = () => {
  1730. if (evaluationCurrent.value >= goodsCommentsInfo.data.list.length) {
  1731. evaluationCurrent.value = goodsCommentsInfo.data.list.length;
  1732. } else {
  1733. evaluationCurrent.value++;
  1734. }
  1735. getEvaluation(evaluationType.value);
  1736. };
  1737. //评价列表当前选择页
  1738. const handleCurrentChange = (val) => {
  1739. evaluationCurrent.value = val;
  1740. getEvaluation(evaluationType.value);
  1741. };
  1742. //获取热销店铺推荐
  1743. const getSalestore = () => {
  1744. let params = {
  1745. storeId: goodsDetail.data.storeInf.storeId,
  1746. sort: 7,
  1747. pageSize: salesPageSize.value,
  1748. current: salesCurrent.value,
  1749. };
  1750. get("v3/goods/front/goods/goodsList", params).then((res) => {
  1751. if (res.state == 200) {
  1752. let result = res.data;
  1753. recommendeData.data = result;
  1754. recommendedList.data = result.list;
  1755. recommendedList.data.map(
  1756. (item) => (item.goodsPrice = new Number(item.goodsPrice).toFixed(2))
  1757. );
  1758. } else {
  1759. ElMessage.error(res.msg);
  1760. }
  1761. });
  1762. };
  1763. //店铺热销列表上一页
  1764. const handlePrevCilickChangeSales = () => {
  1765. if (evaluationCurrent.value == 1) {
  1766. salesCurrent.value = 1;
  1767. } else {
  1768. salesCurrent.value--;
  1769. }
  1770. getSalestore();
  1771. };
  1772. //店铺热销列表下一页
  1773. const handleNextCilickChangeSales = () => {
  1774. if (salesCurrent.value >= goodsCommentsInfo.data.list.length) {
  1775. salesCurrent.value = goodsCommentsInfo.data.list.length;
  1776. } else {
  1777. salesCurrent.value++;
  1778. }
  1779. getSalestore();
  1780. };
  1781. //店铺热销列表当前选择页
  1782. const handleCurrentChangeSales = (val) => {
  1783. salesCurrent.value = val;
  1784. getSalestore();
  1785. };
  1786. //关注店铺及取消关注
  1787. const headercat = ref(null)
  1788. const focusStore = () => {
  1789. if (filtersStore.getLoginFlag) {
  1790. //已登录
  1791. let params = {
  1792. storeIds: goodsDetail.data.storeInf.storeId,
  1793. isCollect: !goodsDetail.data.storeInf.isFollowStore,
  1794. };
  1795. post("v3/member/front/followStore/edit", params).then((res) => {
  1796. if (res.state == 200) {
  1797. goodsDetail.data.storeInf.isFollowStore =!goodsDetail.data.storeInf.isFollowStore;
  1798. proxy.refs.headercat.setfollowStore(goodsDetail.data.storeInf.isFollowStore? "true" : "false")
  1799. if (goodsDetail.data.storeInf.isFollowStore) {
  1800. sldStatEvent({
  1801. behaviorType: "fol",
  1802. storeId: goodsDetail.data.storeInf.storeId,
  1803. });
  1804. }
  1805. }
  1806. });
  1807. } else {
  1808. //未登录提示登录
  1809. return openLoginDialog({
  1810. onRegister: () => {
  1811. router.push({
  1812. path: "/register",
  1813. });
  1814. },
  1815. onForgot: () => {
  1816. router.push({
  1817. path: "/member/login/forget",
  1818. });
  1819. },
  1820. });
  1821. }
  1822. };
  1823. //商品收藏及取消收藏
  1824. const collectGoods = () => {
  1825. if (filtersStore.getLoginFlag) {
  1826. //已登录
  1827. let params = {
  1828. productIds: productId.value,
  1829. isCollect: !goodsDetail.data.followGoods,
  1830. };
  1831. post("v3/member/front/followProduct/edit", params).then((res) => {
  1832. if (res.state == 200) {
  1833. goodsDetail.data.followGoods = !goodsDetail.data.followGoods;
  1834. if (goodsDetail.data.followGoods) {
  1835. sldStatEvent({
  1836. behaviorType: "fav",
  1837. goodsId: goodsDetail.data.goodsId,
  1838. storeId: goodsDetail.data.storeInf.storeId,
  1839. });
  1840. }
  1841. } else {
  1842. ElMessage.error(res.msg);
  1843. }
  1844. });
  1845. } else {
  1846. //未登录提示登录
  1847. return openLoginDialog({
  1848. onRegister: () => {
  1849. router.push({
  1850. path: "/register",
  1851. });
  1852. },
  1853. onForgot: () => {
  1854. router.push({
  1855. path: "/member/login/forget",
  1856. });
  1857. },
  1858. });
  1859. }
  1860. };
  1861. //点击查看全部查看全部的商品规格参数
  1862. const openParameter = () => {
  1863. parameterShow.value = !parameterShow.value;
  1864. };
  1865. //选择商品主图
  1866. const selectDefaultImage = (goodsImgItem, goodsImgIndex) => {
  1867. defaultImage.value = goodsImgItem;
  1868. currentDefaultImage.value = goodsImgIndex;
  1869. };
  1870. //切换商品主图
  1871. const switchDefaultImage = (type) => {
  1872. let defaultImagelength = goodsDetail.data.defaultProduct.goodsPics.length;
  1873. if (type == "left") {
  1874. currentDefaultImage.value--;
  1875. if (currentDefaultImage.value <= 0) {
  1876. currentDefaultImage.value = 0;
  1877. }
  1878. defaultImage.value =
  1879. goodsDetail.data.defaultProduct.goodsPics[currentDefaultImage.value];
  1880. } else {
  1881. currentDefaultImage.value++;
  1882. if (currentDefaultImage.value >= defaultImagelength) {
  1883. currentDefaultImage.value = defaultImagelength - 1;
  1884. }
  1885. defaultImage.value =
  1886. goodsDetail.data.defaultProduct.goodsPics[currentDefaultImage.value];
  1887. }
  1888. goodsPictureListsLeft();
  1889. };
  1890. //图片列表的left移动的距离
  1891. const goodsPictureListsLeft = () => {
  1892. //获取 goods_picture_list 的元素
  1893. let goodsPictureLists = goodsPictureList.value;
  1894. //列表默认显示5张图片
  1895. if (
  1896. goodsDetail.data.defaultProduct.goodsPics.length > 5 &&
  1897. currentDefaultImage.value >= 0
  1898. ) {
  1899. /* 分析找规律:
  1900. 如果有8张图片,点击右键 最大可以向左移动的距离为 8 - 5既3张图,left移动的距离为 3 * - 66px; currentDefaultImage.value从0开始的
  1901. 其中:66px为每一个元素需要每次移动的距离 为图片的宽度与图片之间的间距的和
  1902. currentDefaultImage.value == 4;goodsPictureLists.style.left = 0;
  1903. currentDefaultImage.value == 5;goodsPictureLists.style.left = (5-4) (1) * -66px;
  1904. currentDefaultImage.value == 6;goodsPictureLists.style.left = (5-2) (2) * -66px;
  1905. currentDefaultImage.value == 7;goodsPictureLists.style.left = (5-3) (2) * -66px;
  1906. 点击左键,最大可以向右移动的距离为 8 - 5 即3张
  1907. currentDefaultImage.value == 0;goodsPictureLists.style.left = 0;
  1908. currentDefaultImage.value == 1; goodsPictureLists.style.left = -66px;
  1909. currentDefaultImage.value == 2; goodsPictureLists.style.left = 2 * -66px;
  1910. currentDefaultImage.value == 3; goodsPictureLists.style.left = 3 * -66px;
  1911. */
  1912. if (currentDefaultImage.value > 4) {
  1913. goodsPictureLists.style.left =
  1914. (currentDefaultImage.value - 4) * -66 + "px";
  1915. }
  1916. if (
  1917. currentDefaultImage.value <
  1918. goodsDetail.data.defaultProduct.goodsPics.length - 4
  1919. ) {
  1920. goodsPictureLists.style.left = currentDefaultImage.value * -66 + "px";
  1921. }
  1922. }
  1923. };
  1924. //获取元素距离父元素的顶部及左边的距离
  1925. const offset = (el) => {
  1926. if (process.client) {
  1927. let top = el.offsetTop;
  1928. let left = el.offsetLeft;
  1929. if (el.offsetParent) {
  1930. el = el.offsetParent;
  1931. top += el.offsetTop;
  1932. left += el.offsetLeft;
  1933. }
  1934. return {
  1935. left: left,
  1936. top: top,
  1937. };
  1938. }
  1939. };
  1940. if (process.client) {
  1941. nextTick(() => {
  1942. containerTop.value = offset(container.value).top;
  1943. window.addEventListener("scroll", scrollHandle); //绑定页面滚动事件
  1944. window.addEventListener("click", () => {
  1945. addrDialogVisible.value = false;
  1946. });
  1947. });
  1948. }
  1949. //鼠标移动
  1950. const move = (e) => {
  1951. if (currentDefaultImage.value == -1) {
  1952. return;
  1953. }
  1954. //主图父元素的信息 宽,高
  1955. let previewsBox = previewBox.value;
  1956. let previewBoxWidth = previewsBox.offsetWidth;
  1957. let previewBoxHeight = previewsBox.offsetHeight;
  1958. //主图父元素距离顶部的距离
  1959. let previewsBoxLeft = offset(previewsBox).left;
  1960. let previewsBoxTop = offset(previewsBox).top;
  1961. // 遮罩盒子的信息宽,高
  1962. let masksBox = maskBox.value;
  1963. let maskBoxWidth = masksBox.offsetWidth;
  1964. let maskBoxHeight = masksBox.offsetHeight;
  1965. //鼠标距离屏幕距离
  1966. let moveX = e.clientX;
  1967. let moveY = e.clientY;
  1968. //获取左侧大图父元素的信息
  1969. let zoomsBox = zoomBox.value;
  1970. let zoomBoxWidth = zoomsBox.offsetWidth;
  1971. let zoomBoxHeight = zoomsBox.offsetHeight;
  1972. // 获取大图元素的信息宽,高
  1973. let pictureBigBox = pictureBig.value;
  1974. let pictureBigWidth = pictureBigBox.offsetWidth;
  1975. let pictureBigHeight = pictureBigBox.offsetHeight;
  1976. //获取滚动条的高度
  1977. let scroll = document.documentElement.scrollTop || document.body.scrollTop;
  1978. //主图距离父元素的left及top值
  1979. let left = moveX - previewsBoxLeft - maskBoxWidth / 2;
  1980. let top;
  1981. if (scroll > 0) {
  1982. top = moveY - previewsBoxTop + scroll - maskBoxHeight / 2;
  1983. } else {
  1984. top = moveY - previewsBoxTop - maskBoxHeight / 2;
  1985. }
  1986. //移动限制最大宽度,及最大高度
  1987. let maxWidth = previewBoxWidth - maskBoxWidth;
  1988. let maxHeight = previewBoxHeight - maskBoxHeight;
  1989. left = left < 0 ? 0 : left > maxWidth ? maxWidth : left;
  1990. top = top < 0 ? 0 : top > maxHeight ? maxHeight : top;
  1991. //比列
  1992. let parcentX = left / maxWidth;
  1993. let parcentY = top / maxHeight;
  1994. //遮罩层的定位值
  1995. maskBox.value.style.left = left + "px";
  1996. maskBox.value.style.top = top + "px";
  1997. //大图元素的定位值
  1998. pictureBig.value.style.left =
  1999. parcentX * (zoomBoxWidth - pictureBigWidth) + "px";
  2000. pictureBig.value.style.top =
  2001. parcentY * (zoomBoxHeight - pictureBigHeight) + "px";
  2002. pictureBig.value.style.width =
  2003. (previewBoxWidth / maskBoxWidth) * zoomBoxWidth + "px";
  2004. pictureBig.value.style.height =
  2005. (previewBoxHeight / maskBoxHeight) * zoomBoxHeight + "px";
  2006. };
  2007. //鼠标移出
  2008. const out = () => {
  2009. maskShow.value = false;
  2010. };
  2011. //鼠标移入
  2012. const enter = () => {
  2013. if (currentDefaultImage.value == -1) {
  2014. return;
  2015. }
  2016. addrDialogVisible.value = false;
  2017. maskShow.value = true;
  2018. };
  2019. // 评论区查看图片
  2020. const showImg = (index, img) => {
  2021. imgSource.value = img;
  2022. imgIndex.value = index;
  2023. imgVisible.value = true;
  2024. };
  2025. //分享
  2026. const share = (type) => {
  2027. let title = goodsDetail.data.goodsName; //需要分享的标题,这里取商品名字
  2028. let url = goodsDetail.data.shareLink; //分享的地址,用户点击可以进入到该商品
  2029. let content = goodsDetail.data.goodsBrief; //自定义内容,这里取商品广告词
  2030. let targetUrl = ""; //跳转的url地址
  2031. if (type == "weixin") {
  2032. wxShareCode.value = true;
  2033. //微信
  2034. let canvas = qrcanvas({
  2035. data: url, //二维码内容
  2036. size: 100,
  2037. colorDark: "red",
  2038. });
  2039. let share_wx_qrcode = document.getElementById("share_wx_qrcode");
  2040. if (
  2041. wxShareCode.value &&
  2042. share_wx_qrcode != null &&
  2043. share_wx_qrcode != undefined
  2044. ) {
  2045. document.getElementById("share_wx_qrcode").innerHTML = "";
  2046. document.getElementById("share_wx_qrcode").appendChild(canvas);
  2047. }
  2048. } else if (type == "qzone") {
  2049. wxShareCode.value = false;
  2050. //QQ空间
  2051. targetUrl =
  2052. "https://sns.qzone.qq.com/cgi-bin/qzshare/cgi_qzshare_onekey?title=" +
  2053. encodeURIComponent(title) +
  2054. "&desc=" +
  2055. encodeURIComponent(content) +
  2056. "&summary=" +
  2057. encodeURIComponent(content) +
  2058. "&url=" +
  2059. encodeURIComponent(url);
  2060. window.open(targetUrl, "_blank");
  2061. } else if (type == "sina") {
  2062. wxShareCode.value = false;
  2063. //新浪微博
  2064. targetUrl =
  2065. "https://service.weibo.com/share/share.php?title=" +
  2066. encodeURIComponent(content + "「" + title + "」" + " 点这里" + url);
  2067. window.open(targetUrl, "_blank");
  2068. }
  2069. };
  2070. const refreshInfo = () => {
  2071. history.go(0);
  2072. };
  2073. const updateFllow = (e) => {
  2074. goodsDetail.data.storeInf.isFollowStore = e.state == "true" ? true : false;
  2075. };
  2076. //暴露的变量及方法
  2077. </script>
  2078. <style lang="scss">
  2079. @import "@/assets/style/base.scss";
  2080. @import "@/assets/style/theme.scss";
  2081. @import "@/assets/style/iconfont.css";
  2082. @import "@/assets/style/goodsDetail.scss";
  2083. .popular_list_empty {
  2084. height: 95px;
  2085. font-size: 14px;
  2086. /*font-family: Microsoft YaHei;*/
  2087. font-weight: 400;
  2088. color: #666666;
  2089. }
  2090. .imageBorder {
  2091. border: 1px solid #eee;
  2092. }
  2093. .goods_picture_big {
  2094. border: 1px solid #eee;
  2095. }
  2096. .el-radio__inner:hover {
  2097. border-color: $colorMain;
  2098. }
  2099. .el-radio__input.is-checked .el-radio__inner {
  2100. border-color: $colorMain;
  2101. background: $colorMain;
  2102. }
  2103. .el-radio__input.is-checked + .el-radio__label {
  2104. color: $colorMain;
  2105. }
  2106. .el-radio {
  2107. margin-bottom: 10px;
  2108. display: flex;
  2109. align-items: flex-start;
  2110. white-space: unset;
  2111. margin-right: unset;
  2112. }
  2113. .el-radio-button__inner,
  2114. .el-radio-group {
  2115. /* display: block; */
  2116. line-height: 1;
  2117. vertical-align: middle;
  2118. }
  2119. .el-radio__label {
  2120. font-size: 13px;
  2121. width: 320px;
  2122. overflow: hidden;
  2123. text-overflow: ellipsis;
  2124. display: -webkit-box;
  2125. -webkit-line-clamp: 2;
  2126. -webkit-box-orient: vertical;
  2127. word-break: break-all;
  2128. line-height: 22px;
  2129. margin-top: -5px;
  2130. }
  2131. .evaluationes {
  2132. color: #3b4 !important;
  2133. }
  2134. .sld_goods_detail .goods_htmls .ql-video {
  2135. width: 525px;
  2136. height: 315px;
  2137. }
  2138. .sld_goods_detail .goods_htmls a {
  2139. display: inline-block;
  2140. margin: 5px auto;
  2141. color: #0000ff;
  2142. text-decoration: underline;
  2143. }
  2144. .sld_goods_detail .goods_htmls table {
  2145. border-collapse: collapse;
  2146. padding: 0;
  2147. }
  2148. .sld_goods_detail .goods_htmls td,
  2149. .sld_goods_detail .goods_htmls th {
  2150. border: 1px solid #ddd;
  2151. padding: 5px 10px;
  2152. }
  2153. .sld_goods_detail .goods_htmls ol li,
  2154. .sld_goods_detail .goods_htmls ul li {
  2155. list-style: unset;
  2156. }
  2157. .sld_goods_detail {
  2158. .summary {
  2159. .coupon {
  2160. .el-dialog__header {
  2161. padding-top: 18px;
  2162. padding-bottom: 18px;
  2163. .el-dialog__title {
  2164. color: #333333;
  2165. /*font-family: Microsoft YaHei;*/
  2166. font-weight: bold;
  2167. }
  2168. .el-dialog__close {
  2169. color: #333333;
  2170. font-size: 20px;
  2171. }
  2172. }
  2173. .el-dialog__body {
  2174. background: #f8f8f8;
  2175. }
  2176. }
  2177. }
  2178. }
  2179. .summary-info{
  2180. }
  2181. .mt-20{
  2182. margin-top: 20px;
  2183. }
  2184. .summary_htmls img{
  2185. width: 100%;
  2186. }
  2187. </style>